#pragma once

#include "level.hpp"
#include "message.hpp"
#include <memory>
#include <vector>
#include <sstream>
#include <ctime>
#include <cassert>

namespace superlog{
    // 抽象格式化子项基类
    class FormatItem{
        public:
            using ptr = std::shared_ptr<FormatItem>;
            virtual void format(std::ostream &out, LogMsg &msg) = 0;
    };


    // 派生格式化子项子类 -- 消息，等级，时间，文件名，行号，线程ID，日志器名，换行，其他
    class MsgFormatItem : public FormatItem{
        public:
            void format(std::ostream &out, LogMsg &msg) override{
                out << msg._payload;
            }
    };
    class LevelFormatItem : public FormatItem{
        public:
            void format(std::ostream &out, LogMsg &msg) override{
                out << LogLevel::toString(msg._level);
            }
    };
    class TimeFormatItem : public FormatItem{
        public:
            TimeFormatItem(const std::string &fmt = "%H:%M:%S"): _time_fmt(fmt){}
            void format(std::ostream &out, LogMsg &msg) override{
                struct tm t;
                localtime_r(&msg._ctime, &t);
                char tmp[32] = {0};
                strftime(tmp, 31, _time_fmt.c_str(), &t);
                out << tmp;
            }
        private:
            std::string _time_fmt;  // %H:%M:%S
    };
    class FileFormatItem : public FormatItem{
        public:
            void format(std::ostream &out, LogMsg &msg) override{
                out << msg._file;
            }
    };
    class LineFormatItem : public FormatItem{
        public:
            void format(std::ostream &out, LogMsg &msg) override{
                out << msg._line;
            }
    };
    class ThreadFormatItem : public FormatItem{
        public:
            void format(std::ostream &out, LogMsg &msg) override{
                out << msg._tid;
            }
    };
    class LoggerNameFormatItem : public FormatItem{
        public:
            void format(std::ostream &out, LogMsg &msg) override{
                out << msg._logger;
            }
    };
    class TabFormatItem : public FormatItem{
        public:
            void format(std::ostream &out, LogMsg &msg) override{
                out << "\t";
            }
    };
    class NewLineFormatItem : public FormatItem{
        public:
            void format(std::ostream &out, LogMsg &msg) override{
                out << "\n";
            }
    };
    class OtherFormatItem : public FormatItem {
        public:
            OtherFormatItem(const std::string &str): _str(str){}
            void format(std::ostream &out, LogMsg &msg) override{
                out << _str;
            }
        private:
            std::string _str;
    };


    /*
        %d 表示日期，包含子格式{%H:%M:%S}
        %t 表示线程ID
        %c 表示日志器名称
        %f 表示源码文件名
        %l 表示源码行号
        %p 表示日志等级
        %T 表示制表符缩进
        %m 表示主体消息
        %n 表示换行
    */
    class Formatter{
        public:
            using ptr = std::shared_ptr<Formatter>;
            Formatter(const std::string &pattern = "[%d{%H:%M:%S}][%t][%c][%f:%l][%p]%T%m%n")
                :_pattern(pattern){
                    assert(parsePattern());
                }

            // 对msg进行格式化
            void format(std::ostream &out, LogMsg &msg){
                for (auto &item : _items){
                    item->format(out, msg);
                }
            }
            
            std::string format(LogMsg &msg){
                std::stringstream ss;
                format(ss, msg);
                return ss.str();
            }

        private:
            // 对格式化规则字符串进行解析
            bool parsePattern(){
                // 1. 对格式化规则字符串进行解析
                std::vector<std::pair<std::string, std::string>> fmt_order;
                size_t pos = 0;
                std::string key, val;

                while(pos < _pattern.size())
                {
                    // 1.1 处理原始字符--判断是否是%，不是就是原始字符
                    if (_pattern[pos] != '%'){
                        val.push_back(_pattern[pos]);
                        pos++;
                        continue;
                    }
                    // 能走下来，就代表pos位置就是%字符
                    // 1.2 处理双%的情况，处理为一个原始%字符
                    if (pos + 1 < _pattern.size() && _pattern[pos + 1] == '%'){
                        val.push_back('%'); 
                        pos += 2;
                        continue;
                    }
                    // 走到这里表示，这里是一个格式化字符，原始字符串处理完毕
                    if (val.empty() == false){
                        fmt_order.push_back(std::make_pair("", val));
                        val.clear();
                    }

                    // 走到这里，开始处理格式化字符，这时pos指向%位置
                    pos++;
                    if (pos == _pattern.size()){
                        std::cout << "%之后，没有对应的格式化字符！\n";
                        return false;
                    }
                    key = _pattern[pos];

                    // 1.3 判断pos后是否有{
                    pos++;

                    if (pos < _pattern.size() && _pattern[pos] == '{'){
                        pos += 1; // 这时，pos指向子规则的起始位置
                        while(pos < _pattern.size() && _pattern[pos] != '}'){
                            val.push_back(_pattern[pos++]);
                        }
                        // 走到末尾跳出了循环，没有遇到}，代表格式是错误的
                        if (pos == _pattern.size()) {
                            std::cout << "子规则匹配出错！\n";
                            return false;   // 没有找到'}'
                        }
                        // 这时pos指向的是}位置，向后走一步就走出了子规则，走到了下次处理的新位置
                        pos++;  
                    }
                    fmt_order.push_back(std::make_pair(key, val));
                    key.clear();
                    val.clear();
                }
                // 2. 根据解析得到的数据格式化子项数组成员
                for (auto &it : fmt_order){
                    _items.push_back(createItem(it.first, it.second));
                }
                return true;
            }

            // 根据不同的格式化字符，创建不同的格式化子项
            FormatItem::ptr createItem(const std::string &key, const std::string &val){
                if (key == "d") return std::make_shared<TimeFormatItem>(val);
                if (key == "t") return std::make_shared<ThreadFormatItem>();
                if (key == "c") return std::make_shared<LoggerNameFormatItem>();
                if (key == "f") return std::make_shared<FileFormatItem>();
                if (key == "l") return std::make_shared<LineFormatItem>();
                if (key == "p") return std::make_shared<LevelFormatItem>();
                if (key == "T") return std::make_shared<TabFormatItem>();
                if (key == "m") return std::make_shared<MsgFormatItem>();
                if (key == "n") return std::make_shared<NewLineFormatItem>();
                if (key.empty()) return std::make_shared<OtherFormatItem>(val);
                std::cout << "没有对应的格式化字符: %" << key << std::endl;
                abort();
                return FormatItem::ptr();
            }
        private:
            std::string _pattern;   // 格式化规则字符串
            std::vector<FormatItem::ptr> _items;
    };
}